home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevbj10.c < prev    next >
C/C++ Source or Header  |  1997-03-03  |  13KB  |  386 lines

  1. /* Copyright (C) 1990, 1995, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevbj10.c */
  20. /* Canon Bubble Jet BJ-10e and BJ200 printer driver */
  21. #include "gdevprn.h"
  22.  
  23. /*
  24.  * The following is taken from the BJ200 Programmer's manual.  The top
  25.  * margin is 3mm (0.12"), and the bottom margin is 6.4mm (0.25").  The
  26.  * left and right margin depend on the type of paper -- US letter or
  27.  * A4 -- but ultimately rest on a print width of 203.2mm (8").  For letter
  28.  * paper, the left margin (and hence the right) is 6.4mm (0.25"), while
  29.  * for A4 paper, both are 3.4mm (0.13").
  30.  *
  31.  * The bottom margin requires a bit of care.  The image is printed
  32.  * as strips, each about 3.4mm wide.  We can only attain the bottom 
  33.  * margin if the final strip coincides with it.  Note that each strip
  34.  * is generated using only 48 of the available 64 jets, and the absence
  35.  * of those bottom 16 jets makes our bottom margin, in effect, about
  36.  * 1.1mm (0.04") larger.
  37.  *
  38.  * The bj200 behaves, in effect, as though the origin were at the first
  39.  * printable position, rather than the top left corner of the page, so
  40.  * we add a translation to the initial matrix to compensate for this.
  41.  *
  42.  * Except for the details of getting the margins correct, the bj200 is
  43.  * no different from the bj10e, and uses the same routine to print each
  44.  * page.
  45.  *
  46.  * NOTE:  The bj200 has a DIP switch called "Text scale mode" and if
  47.  * set, it allows the printer to get 66 lines on a letter-sized page
  48.  * by reducing the line spacing by a factor of 14/15.  If this DIP
  49.  * switch is set, the page image printed by ghostscript will also be
  50.  * similarly squeezed.  Thus text scale mode is something ghostscript
  51.  * would like to disable.
  52.  *
  53.  * According to the bj200 manual, which describes the bj10 commands,
  54.  * the printer can be reset either to the settings determined by the
  55.  * DIP switches, or to the factory defaults, and then some of those
  56.  * settings can be specifically overriden.  Unfortunately, the text
  57.  * scale mode and horizontal print position (for letter vs A4 paper)
  58.  * can not be overriden.  On my bj200, the factory settings are for
  59.  * no text scaling and letter paper, thus using the factory defaults
  60.  * also implies letter paper.  I don't know if this is necessarily
  61.  * true for bj200's sold elsewhere, or for other printers that use
  62.  * the same command set.
  63.  *
  64.  * If your factory defaults are in fact the same, you can compile
  65.  * the driver with USE_FACTORY_DEFAULTS defined, in which case the
  66.  * printer will be reset to the factory defaults for letter paper,
  67.  * and reset to the DIP switch settings for A4 paper.  In this case,
  68.  * with letter-sized paper, the text scale mode will be disabled.
  69.  * Further, by leaving the horizontal print position DIP switch set
  70.  * for A4 paper, gs will be able to print on either A4 or letter
  71.  * paper without changing the DIP switch.  Since it's not clear that
  72.  * the factory defaults are universal, the default behaviour is not
  73.  * to define USE_FACTORY_DEFAULTS, and the printer will always be
  74.  * reset to the DIP switch defaults.
  75.  */
  76.  
  77. #define BJ200_TOP_MARGIN        0.12
  78. #define BJ200_BOTTOM_MARGIN        0.29
  79. #define BJ200_LETTER_SIDE_MARGIN    0.25
  80. #define BJ200_A4_SIDE_MARGIN        0.13
  81.  
  82. private dev_proc_open_device(bj200_open);
  83.  
  84. private dev_proc_print_page(bj10e_print_page);
  85.  
  86. private gx_device_procs prn_bj200_procs =
  87.   prn_procs(bj200_open, gdev_prn_output_page, gdev_prn_close);
  88.  
  89. gx_device_printer far_data gs_bj200_device =
  90.   prn_device(prn_bj200_procs, "bj200",
  91.     DEFAULT_WIDTH_10THS,
  92.     DEFAULT_HEIGHT_10THS,
  93.     360,                /* x_dpi */
  94.     360,                /* y_dpi */
  95.     0, 0, 0, 0,            /* margins filled in by bj200_open */
  96.     1, bj10e_print_page);
  97.  
  98. /*
  99.  * (<simon@pogner.demon.co.uk>, aka <sjwright@cix.compulink.co.uk>):
  100.  * My bj10ex, which as far as I can tell is just like a bj10e, works
  101.  * fine with the bj200 setup here.
  102.  */
  103.  
  104. gx_device_printer far_data gs_bj10e_device =
  105.   prn_device(prn_bj200_procs, "bj10e",
  106.     DEFAULT_WIDTH_10THS,
  107.     DEFAULT_HEIGHT_10THS,
  108.     360,                /* x_dpi */
  109.     360,                /* y_dpi */
  110.     0,0,0,0,            /* margins */
  111.     1, bj10e_print_page);
  112.  
  113. /*
  114.  * Notes on the BJ10e/BJ200 command set.
  115.  *
  116.  
  117. According to the BJ200 manual, the "set initial condition" sequence (ESC [
  118. K) has 2 bytes which can override the DIP switches -- these are the last 2
  119. bytes.  Several bits are listed as "reserved" -- one or more may possibly
  120. affect the sheet feeder.  The first is referred to as <P1>, with the
  121. following meaning:
  122.                 1        0
  123. bit 7    ignore/process P1    ignore        process
  124. bit 6    reserved
  125. bit 5    alarm            disabled    enabled
  126. bit 4    automatic CR        CR+LF        CR
  127. bit 3    automatic LF        CR+LF        LF
  128. bit 2    page length        12 inches    11 inches
  129. bit 1    style for zero        slashed        not slashed
  130. bit 0    character set        set 2        set 1
  131.  
  132. The last byte is <P2>, with the following meaning:
  133.                 1        0
  134. bit 7    ignore/process P2    ignore        process
  135. bit 6    code page        850        437
  136. bit 5    reserved
  137. bit 4    reserved
  138. bit 3    reserved
  139. bit 2    reserved
  140. bit 1    reserved
  141. bit 0    reserved
  142.  
  143. The automatic CR setting is important to gs, but the rest shouldn't matter
  144. (gs doesn't print characters or send LF, and it explicitly sets the page
  145. length).  The sequence ESC 5 <n> controls automatic CR -- if <n> is 0x00,
  146. it is turned off (CR only) and if <n> is 0x01, it is turned on (CR + LF).
  147. So we do following: Change the initialization string to so that the last 2
  148. of the 9 bytes are \200 rather than \000.  Then add
  149.     |* Turn off automatic carriage return, otherwise we get line feeds. *|
  150.     fwrite("\0335\000", 1, 3, prn_stream);
  151. after the initialization.  (Actually, instead of setting the last 2 bytes
  152. to \200, we suppress them altogether by changing the byte count from \004
  153. to \002 (the byte count is the 4th (low 8 bits) and 5th (high 8 bits) bytes
  154. in the initialization sequence).)
  155.  
  156. */
  157.  
  158. /* ------ Internal routines ------ */
  159.  
  160. /* Open the printer, and set the margins. */
  161. private int
  162. bj200_open(gx_device *pdev)
  163. {
  164.     /* Change the margins according to the paper size.
  165.        The top and bottom margins don't seem to depend on the
  166.        page length, but on the paper handling mechanism;
  167.        The side margins do depend on the paper width, as the
  168.        printer centres the 8" print line on the page. */
  169.  
  170.     static const float a4_margins[4] =
  171.      {    BJ200_A4_SIDE_MARGIN, BJ200_BOTTOM_MARGIN,
  172.         BJ200_A4_SIDE_MARGIN, BJ200_TOP_MARGIN
  173.      };
  174.     static const float letter_margins[4] =
  175.      {    BJ200_LETTER_SIDE_MARGIN, BJ200_BOTTOM_MARGIN,
  176.         BJ200_LETTER_SIDE_MARGIN, BJ200_TOP_MARGIN
  177.      };
  178.  
  179.     gx_device_set_margins(pdev,
  180.         (pdev->width / pdev->x_pixels_per_inch <= 8.4 ?
  181.          a4_margins : letter_margins),
  182.         true);
  183.     return gdev_prn_open(pdev);
  184. }
  185.  
  186. /* Send the page to the printer. */
  187. private int
  188. bj10e_print_page(gx_device_printer *pdev, FILE *prn_stream)
  189. {    int line_size = gx_device_raster((gx_device *)pdev, 0);
  190.     int xres = pdev->x_pixels_per_inch;
  191.     int yres = pdev->y_pixels_per_inch;
  192.     int mode = (yres == 180 ?
  193.             (xres == 180 ? 11 : 12) :
  194.             (xres == 180 ? 14 : 16));
  195.     int bytes_per_column = (yres == 180) ? 3 : 6;
  196.     int bits_per_column = bytes_per_column * 8;
  197.     int skip_unit = bytes_per_column * 3;
  198.     byte *in = (byte *)gs_malloc(8, line_size, "bj10e_print_page(in)");
  199.     byte *out = (byte *)gs_malloc(bits_per_column, line_size, "bj10e_print_page(out)");
  200.     int lnum = 0;
  201.     int skip = 0;
  202.     int code = 0;
  203.     int last_row = dev_print_scan_lines(pdev);
  204.     int limit = last_row - bits_per_column;
  205.  
  206.     if ( in == 0 || out == 0 )
  207.     {    code = gs_note_error(gs_error_VMerror);
  208.         goto fin;
  209.     }
  210.  
  211.     /* Initialize the printer. */
  212. #ifdef USE_FACTORY_DEFAULTS
  213.     /* Check for U.S. letter vs. A4 paper. */
  214.     fwrite(( pdev->width / pdev->x_pixels_per_inch <= 8.4 ?
  215.         "\033[K\002\000\000\044"    /*A4--DIP switch defaults*/ :
  216.         "\033[K\002\000\004\044"    /*letter--factory defaults*/ ),
  217.            1, 7, prn_stream);
  218. #else
  219.     fwrite("\033[K\002\000\000\044", 1, 7, prn_stream);
  220. #endif
  221.  
  222.     /* Turn off automatic carriage return, otherwise we get line feeds. */
  223.     fwrite("\0335\000", 1, 3, prn_stream);
  224.  
  225.     /* Set vertical spacing. */
  226.     fwrite("\033[\\\004\000\000\000", 1, 7, prn_stream);
  227.     fputc(yres & 0xff, prn_stream);
  228.     fputc(yres >> 8, prn_stream);
  229.  
  230.     /* Set the page length.  This is the printable length, in inches. */
  231.     fwrite("\033C\000", 1, 3, prn_stream);
  232.     fputc((last_row + yres - 1)/yres, prn_stream);
  233.  
  234.     /* Transfer pixels to printer.  The last row we can print is defined
  235.        by "last_row".  Only the bottom of the print head can print at the
  236.        bottom margin, and so we align the final printing pass.  The print
  237.        head is kept from moving below "limit", which is exactly one pass
  238.        above the bottom margin.  Once it reaches this limit, we make our
  239.        final printing pass of a full "bits_per_column" rows. */
  240.     while ( lnum < last_row )
  241.        {    
  242.         byte *in_data;
  243.         byte *in_end = in + line_size;
  244.         byte *out_beg = out;
  245.         byte *out_end = out + bytes_per_column * pdev->width;
  246.         byte *outl = out;
  247.         int bnum;
  248.  
  249.         /* Copy 1 scan line and test for all zero. */
  250.         code = gdev_prn_get_bits(pdev, lnum, in, &in_data);
  251.         if ( code < 0 ) goto xit;
  252.         /* The mem... or str... functions should be faster than */
  253.         /* the following code, but all systems seem to implement */
  254.         /* them so badly that this code is faster. */
  255.            {    register const long *zip = (const long *)in_data;
  256.             register int zcnt = line_size;
  257.             register const byte *zipb;
  258.             for ( ; zcnt >= 4 * sizeof(long); zip += 4, zcnt -= 4 * sizeof(long) )
  259.                {    if ( zip[0] | zip[1] | zip[2] | zip[3] )
  260.                     goto notz;
  261.                }
  262.             zipb = (const byte *)zip;
  263.             while ( --zcnt >= 0 )
  264.                {
  265.                 if ( *zipb++ )
  266.                     goto notz;
  267.                }
  268.             /* Line is all zero, skip */
  269.             lnum++;
  270.             skip++;
  271.             continue;
  272. notz:            ;
  273.            }
  274.  
  275.         /* Vertical tab to the appropriate position.  Note here that
  276.            we make sure we don't move below limit. */
  277.         if ( lnum > limit )
  278.             {    skip -= (lnum - limit);
  279.             lnum = limit;
  280.             }
  281.         while ( skip > 255 )
  282.            {    fputs("\033J\377", prn_stream);
  283.             skip -= 255;
  284.            }
  285.         if ( skip )
  286.             fprintf(prn_stream, "\033J%c", skip);
  287.  
  288.         /* If we've printed as far as "limit", then reset "limit"
  289.            to "last_row" for the final printing pass. */
  290.         if ( lnum == limit )
  291.             limit = last_row;
  292.         skip = 0;
  293.  
  294.         /* Transpose in blocks of 8 scan lines. */
  295.         for ( bnum = 0; bnum < bits_per_column; bnum += 8 )
  296.            {    int lcnt = min(8, limit - lnum);
  297.             byte *inp = in;
  298.             byte *outp = outl;
  299.                lcnt = gdev_prn_copy_scan_lines(pdev,
  300.                 lnum, in, lcnt * line_size);
  301.             if ( lcnt < 0 )
  302.                {    code = lcnt;
  303.                 goto xit;
  304.                }
  305.             if ( lcnt < 8 )
  306.                 memset(in + lcnt * line_size, 0,
  307.                        (8 - lcnt) * line_size);
  308.             for ( ; inp < in_end; inp++, outp += bits_per_column )
  309.                {    gdev_prn_transpose_8x8(inp, line_size,
  310.                     outp, bytes_per_column);
  311.                }
  312.             outl++;
  313.             lnum += lcnt;
  314.             skip += lcnt;
  315.            }
  316.  
  317.         /* Send the bits to the printer.  We alternate horizontal
  318.            skips with the data.  The horizontal skips are in units
  319.            of 1/120 inches, so we look at the data in groups of
  320.            3 columns, since 3/360 = 1/120, and 3/180 = 2/120.  */
  321.         outl = out;
  322.         do
  323.            {    int count;
  324.             int n;
  325.             byte *out_ptr;
  326.  
  327.             /* First look for blank groups of columns. */
  328.             while(outl < out_end)
  329.                {    n = count = min(out_end - outl, skip_unit);
  330.                 out_ptr = outl;
  331.                 while ( --count >= 0 )
  332.                    {    if ( *out_ptr++ )
  333.                         break;
  334.                    }
  335.                 if ( count >= 0 )
  336.                     break;
  337.                 else
  338.                     outl = out_ptr;
  339.                }
  340.             if (outl >= out_end)
  341.                 break;
  342.             if (outl > out_beg)
  343.                {    count = (outl - out_beg) / skip_unit;
  344.                 if ( xres == 180 ) count <<= 1;
  345.                 fprintf(prn_stream, "\033d%c%c",
  346.                     count & 0xff, count >> 8);
  347.                }
  348.  
  349.             /* Next look for non-blank groups of columns. */
  350.             out_beg = outl;
  351.             outl += n;
  352.             while(outl < out_end)
  353.                {    n = count = min(out_end - outl, skip_unit);
  354.                 out_ptr = outl;
  355.                 while ( --count >= 0 )
  356.                    {    if ( *out_ptr++ )
  357.                         break;
  358.                    }
  359.                 if ( count < 0 )
  360.                     break;
  361.                 else
  362.                     outl += n;
  363.                }
  364.             count = outl - out_beg + 1;
  365.             fprintf(prn_stream, "\033[g%c%c%c",
  366.                 count & 0xff, count >> 8, mode);
  367.             fwrite(out_beg, 1, count - 1, prn_stream);
  368.             out_beg = outl;
  369.             outl += n;
  370.            }
  371.         while ( out_beg < out_end );
  372.  
  373.         fputc('\r', prn_stream);
  374.        }
  375.  
  376.     /* Eject the page */
  377. xit:    fputc(014, prn_stream);    /* form feed */
  378.     fflush(prn_stream);
  379. fin:    if ( out != 0 )
  380.         gs_free((char *)out, bits_per_column, line_size,
  381.             "bj10e_print_page(out)");
  382.     if ( in != 0 )
  383.         gs_free((char *)in, 8, line_size, "bj10e_print_page(in)");
  384.     return code;
  385. }
  386.